package com.atguigu.daijia.order.testLock;


import org.apache.commons.lang3.StringUtils;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.stereotype.Service;

import java.util.Arrays;
import java.util.UUID;
import java.util.concurrent.TimeUnit;

/**
 * 实现TestService接口的服务类
 */
@Service
public class TestServiceImpl implements TestService {
    // 注入Redis模板用于操作Redis
    @Autowired
    private StringRedisTemplate redisTemplate;

    // 注入Redisson客户端用于分布式锁
    @Autowired
    private RedissonClient redissonClient;

    /**
     * 使用Redisson实现分布式锁的测试方法
     * 开发步骤：
     * 1. 使用RedissonClient客户端对象创建锁对象
     * 2. 调用获取锁方法
     * 3. 执行业务逻辑
     * 4. 将锁释放
     */
    public void testLock() {
        // 0. 创建锁对象
        RLock lock = redissonClient.getLock("lock1");

        // 0.1 尝试加锁
        // lock() 阻塞等待一直到获取锁,默认锁有效期30s
        lock.lock();

        // 1. 先从Redis中通过key num获取值  key提前手动设置 num 初始值：0
        String value = redisTemplate.opsForValue().get("num");
        // 2. 如果值为空则非法直接返回即可
        if (StringUtils.isBlank(value)) {
            return;
        }
        // 3. 对num值进行自增加一
        int num = Integer.parseInt(value);
        redisTemplate.opsForValue().set("num", String.valueOf(++num));

        // 4. 将锁释放
        lock.unlock();
    }

    /*   *//**
     * 采用SpringDataRedis实现分布式锁的测试方法
     * 原理：执行业务方法前先尝试获取锁（setnx存入key val），如果获取锁成功再执行业务代码，业务执行完毕后将锁释放(del key)
     *//*
    @Override
    public void testLock() {
        // 0. 先尝试获取锁 setnx key val
        // 问题：锁可能存在线程间相互释放
        // Boolean flag = stringRedisTemplate.opsForValue().setIfAbsent("lock", "lock", 10, TimeUnit.SECONDS);
        // 解决：锁值设置为uuid
        String uuid = UUID.randomUUID().toString();
        Boolean flag = redisTemplate.opsForValue().setIfAbsent("lock", uuid, 10, TimeUnit.SECONDS);

        if (flag) {
            // 获取锁成功，执行业务代码
            // 1. 先从Redis中通过key num获取值  key提前手动设置 num 初始值：0
            String value = redisTemplate.opsForValue().get("num");
            // 2. 如果值为空则非法直接返回即可
            if (StringUtils.isBlank(value)) {
                return;
            }
            // 3. 对num值进行自增加一
            int num = Integer.parseInt(value);
            redisTemplate.opsForValue().set("num", String.valueOf(++num));

            // 4. 将锁释放 判断uuid
            // 问题：删除操作缺乏原子性。
            // if(uuid.equals(stringRedisTemplate.opsForValue().get("lock"))){ //线程一：判断是满足是当前线程锁的值
            //    //条件满足，此时锁正好到期，redis锁自动释放了线程2获取锁成功，线程1将线程2的锁删除
            //    stringRedisTemplate.delete("lock");
            // }
            // 解决：Redis执行Lua脚本保证原子，Lua脚本执行会作为一个整体执行

            // 执行脚本参数 参数1：脚本对象封装Lua脚本，参数二：Lua脚本中需要key参数（KEYS[i]）  参数三：Lua脚本中需要参数值 ARGV[i]
            // 4.1 先创建脚本对象 DefaultRedisScript泛型脚本语言返回值类型 Long 0：失败 1：成功
            DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
            // 4.2 设置脚本文本
            String script = "if redis.call(\"get\",KEYS[1]) == ARGV[1]\n" +
                    "then\n" +
                    "    return redis.call(\"del\",KEYS[1])\n" +
                    "else\n" +
                    "    return 0\n" +
                    "end";
            redisScript.setScriptText(script);
            // 4.3 设置响应类型
            redisScript.setResultType(Long.class);
            redisTemplate.execute(redisScript, Arrays.asList("lock"), uuid);
        } else {
            try {
                // 睡眠
                Thread.sleep(100);
                // 自旋重试
                this.testLock();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }*/

    // 另一个分布式锁测试方法
    public void testLock2() {
        String uuid = UUID.randomUUID().toString();
        // 0. 先尝试获取锁 setnx key val
        Boolean flag = redisTemplate.opsForValue().setIfAbsent("lock", uuid, 2, TimeUnit.SECONDS);
        if (flag) {
            // 获取锁成功，执行业务代码
            // 1. 先从Redis中通过key num获取值  key提前手动设置 num 初始值：0
            String value = redisTemplate.opsForValue().get("num");
            // 2. 如果值为空则非法直接返回即可
            if (StringUtils.isBlank(value)) {
                return;
            }
            // 3. 对num值进行自增加一
            int num = Integer.parseInt(value);
            redisTemplate.opsForValue().set("num", String.valueOf(++num));
            if (uuid.equals((String) redisTemplate.opsForValue().get("lock"))) {
                // 4. 将锁释放
                redisTemplate.delete("lock");
            }
        } else {
            try {
                Thread.sleep(100);
                this.testLock();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    // 同步方法实现的测试锁方法
    public synchronized void testLock1() {
        // 查询Redis中的num值
        String value = (String) this.redisTemplate.opsForValue().get("num");
        // 没有该值return
        if (StringUtils.isBlank(value)) {
            return;
        }
        // 有值就转成int
        int num = Integer.parseInt(value);
        // 把Redis中的num值+1
        this.redisTemplate.opsForValue().set("num", String.valueOf(++num));
    }
}
